

# УНИВЕРЗИТЕТ У НОВОМ САДУ ФАКУЛТЕТ ТЕХНИЧКИХ НАУКА



# Димитрије Ћук

# Од изворног С кода до извршне бинарне слике - компилација и распоред у меморији микроконтролера

Дипломски рад

Нови Сад 2025.



# УНИВЕРЗИТЕТ У НОВОМ САДУ ● **ФАКУЛТЕТ ТЕХНИЧКИХ НАУКА**21000 НОВИ САД, Трг Доситеја Обрадовића 6

# КЉУЧНА ДОКУМЕНТАЦИЈСКА ИНФОРМАЦИЈА

| Редни број, <b>РБР</b> :                                                         |                  |                                                                                                                                                                                                                                                                                                                |                       |  |
|----------------------------------------------------------------------------------|------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|-----------------------|--|
| Идентификациони број,                                                            | ИБР:             |                                                                                                                                                                                                                                                                                                                |                       |  |
| Тип документације, <b>тд</b> :                                                   |                  | Монографска публикација                                                                                                                                                                                                                                                                                        |                       |  |
| Тип записа, <b>Т3</b> :                                                          |                  | Текстуални штампани примерак                                                                                                                                                                                                                                                                                   |                       |  |
| Врста рада, <b>ВР</b> :                                                          |                  | Дипломски рад                                                                                                                                                                                                                                                                                                  |                       |  |
| Аутор, <b>АУ</b> :                                                               |                  | Димитрије Ћук                                                                                                                                                                                                                                                                                                  |                       |  |
| Ментор, <b>МН</b> :                                                              |                  | проф. др Дарко Марчетић                                                                                                                                                                                                                                                                                        |                       |  |
| Наслов рада, <b>НР</b> :                                                         |                  | Од изворног С кода до извршне бинарне слике - компилација и распоред у<br>меморији микроконтролера                                                                                                                                                                                                             |                       |  |
| Језик публикације, <b>ЈП</b> :                                                   |                  | српски                                                                                                                                                                                                                                                                                                         |                       |  |
| Језик извода, <b>ЈИ</b> :                                                        |                  | српски                                                                                                                                                                                                                                                                                                         |                       |  |
| Земља публиковања, <b>3П</b> :                                                   |                  | Република Србија                                                                                                                                                                                                                                                                                               |                       |  |
| Уже географско подруч                                                            | је, <b>УГП</b> : | АП Војводина                                                                                                                                                                                                                                                                                                   |                       |  |
| Година, <b>ГО</b> :                                                              |                  | 2024.                                                                                                                                                                                                                                                                                                          |                       |  |
| Издавач, <b>ИЗ</b> :                                                             |                  | Ауторски репринт                                                                                                                                                                                                                                                                                               |                       |  |
| Место и адреса, <b>МА</b> :                                                      |                  | Факултет техничких наука, 21000 Нови Сад, Трг Д                                                                                                                                                                                                                                                                | Јоситеја Обрадовића 6 |  |
| Физички опис рада, ФО:<br>(поглавља/страна/ цитата/табела/спика/графика/прилога) |                  | (9/72/15/2/35/0/0)                                                                                                                                                                                                                                                                                             |                       |  |
| Научна област, <b>НО</b> :                                                       |                  | Електротехничко и рачунарско инжењерство                                                                                                                                                                                                                                                                       |                       |  |
| Научна дисциплина, <b>НД</b> :                                                   |                  | Рачунарске науке и уграђени системи                                                                                                                                                                                                                                                                            |                       |  |
| Предметна одредница/Кључне речи, <b>ПО</b> :                                     |                  | Меморија у микроконтролеру, компајлер и компајлирање, линкер и линковање                                                                                                                                                                                                                                       |                       |  |
| удк                                                                              |                  |                                                                                                                                                                                                                                                                                                                |                       |  |
| Чува се, <b>ЧУ</b> :                                                             |                  | Библиотека ФТН, Трг Доситеја Обрадовића 6, Нови Сад                                                                                                                                                                                                                                                            |                       |  |
| Важна напомена, <b>ВН</b> :                                                      |                  |                                                                                                                                                                                                                                                                                                                |                       |  |
| Извод, <b>ИЗ</b> :                                                               |                  | С обзиром да у микроконтролеру не постоји оперативни систем који управља меморијом, потребно је конфигурисати линкер кроз линкер директиву, односно линкер скрипту, на начин да уважи физичку расподелу меморије по врстама и адресама, на основу техничког листа, тј. спецификације коју произвођач доставља. |                       |  |
|                                                                                  |                  | Дигитално потписивање је процес који осигурава аутентичност и интегритет података коришћењем криптографије. Користи се асиметрична енкрипција која подразумева постојање приватног и јавног кључа.                                                                                                             |                       |  |
| Датум прихватања теме                                                            | <br>e, ДП:       | 10.09.2024.                                                                                                                                                                                                                                                                                                    |                       |  |
| Датум одбране, <b>ДО</b> :                                                       |                  | 27.09.2024.                                                                                                                                                                                                                                                                                                    |                       |  |
| Чланови комисије, <b>КО</b> : Председник:                                        |                  | др Владимир Поповић, доц., ФТН Нови Сад                                                                                                                                                                                                                                                                        |                       |  |
|                                                                                  | Члан:            | др Стеван Цветићанин, ван. проф., ФТН Нови<br>Сад                                                                                                                                                                                                                                                              |                       |  |
|                                                                                  | Члан:            |                                                                                                                                                                                                                                                                                                                | Потпис ментора        |  |
|                                                                                  | Члан:<br>        |                                                                                                                                                                                                                                                                                                                |                       |  |
|                                                                                  | Члан, ментор:    | др Дарко Марчетић, ред. проф., ФТН Нови Сад                                                                                                                                                                                                                                                                    |                       |  |

Образац **Q2.HA.06-05** - Издање 1



#### UNIVERSITY OF NOVI SAD • FACULTY OF TECHNICAL SCIENCES

21000 NOVI SAD, Trg Dositeja Obradovića 6

#### **KEY WORDS DOCUMENTATION**

| Accession number, ANO:                                                           |             |                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                               |                |  |
|----------------------------------------------------------------------------------|-------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|----------------|--|
| Identification number, INO:                                                      |             |                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                               |                |  |
| Document type, <b>DT</b> :                                                       |             | Monographic publication                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                       |                |  |
| Type of record, <b>TR</b> :                                                      |             | Textual Printed Material                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                      |                |  |
| Contents code, CC:                                                               |             | Bachelor thesis                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                               |                |  |
| Author, <b>AU</b> :                                                              |             | Dimitrije Ćuk                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                 |                |  |
| Mentor, <b>MN</b> :                                                              |             | prof. dr Darko Marčetić                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                       |                |  |
| Title, <b>TI</b> :                                                               |             | Memory Mapping in a Microcontroller and the Use of Digital Signatures                                                                                                                                                                                                                                                                                                                                                                                                                                                                         |                |  |
| Language of text, <b>LT</b> :                                                    |             | Serbian                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                       |                |  |
| Language of abstract, <b>L</b>                                                   | <b>A</b> :  | Serbian                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                       |                |  |
| Country of publication, <b>C</b>                                                 | CP:         | Republic of Serbia                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                            |                |  |
| Locality of publication, L                                                       | <b>.</b> P: | AP of Vojvodina                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                               |                |  |
| Publication year, <b>PY</b> :                                                    |             | 2024.                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                         |                |  |
| Publisher, <b>PB</b> :                                                           |             | Author's reprint                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                              |                |  |
| Publication place, <b>PP</b> :                                                   |             | Faculty of technical sciences, 21000 Novi Sad, Trg Dositeja Obradovića 6                                                                                                                                                                                                                                                                                                                                                                                                                                                                      |                |  |
| Physical description, PD: (chapters/pages/ref./tables/pictures/graphs/appendixes |             | (9/72/15/2/35/0/0)                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                            |                |  |
| Scientific field, <b>SF</b> :                                                    |             | Electrical and computer engineering                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                           |                |  |
| Scientific discipline, <b>SD</b> :                                               |             | Computer science and embedded systems                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                         |                |  |
| Subject/Key words, <b>S/KW</b> :                                                 |             | From Source C Code to Executable Binary Image – Compilation and Memory Layout in Microcontrollers                                                                                                                                                                                                                                                                                                                                                                                                                                             |                |  |
| UC                                                                               |             |                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                               |                |  |
| Holding data, <b>HD</b> :                                                        |             | Library of Faculty of technical sciences, Trg Dositeja Obradovića 6, Novi Sad                                                                                                                                                                                                                                                                                                                                                                                                                                                                 |                |  |
| Note, <b>N</b> : Abstract, <b>AB</b> :                                           |             | Since there is no operating system in a microcontroller to manage memory, it is necessary to configure the linker through a linker directive or linker script in such a way that it takes into account the physical distribution of memory by type and address, based on the datasheet or specification provided by the manufacturer.  Digital signing is a process that ensures the authenticity and integrity of data through cryptography. It uses asymmetric encryption, which involves the existence of both a private and a public key. |                |  |
| Accepted by the Scientific Board on, <b>ASB</b> :                                |             | 10.09.2024.                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                   |                |  |
| Defended on, <b>DE</b> :                                                         |             | 27.09.2024.                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                   |                |  |
| Defended Board, <b>DB</b> : President:                                           |             | dr Vladimir Popović, assist. prof., FTN Novi Sad                                                                                                                                                                                                                                                                                                                                                                                                                                                                                              |                |  |
|                                                                                  | Member:     | dr Stevan Cvetićanin, assoc. prof., FTN Novi Sad                                                                                                                                                                                                                                                                                                                                                                                                                                                                                              |                |  |
|                                                                                  | Member:     |                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                               | Menthor's sign |  |
|                                                                                  | Member:     |                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                               |                |  |
| Member,<br>mentor:                                                               |             | dr Darko Marčetić, full prof., FTN Novi Sad                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                   |                |  |

Образац **Q2.HA.06-05** - Издање 1



# УНИВЕРЗИТЕТ У НОВОМ САДУ ● ФАКУЛТЕТ ТЕХНИЧКИХ НАУКА 21000 НОВИ САД, Трг Доситеја Обрадовића 6

| Број:<br>012-40/1732 |
|----------------------|
| Датум:               |
| 10.09.2024.          |

# ЗАДАТАК ЗА ДИПЛОМСКИ РАД

(Податке уноси предметни наставник - ментор)

| СТУДИЈСКИ<br>ПРОГРАМ:                  |                         | E1 – енергетика, електроника, телекомуникације |                         |                      |                         |
|----------------------------------------|-------------------------|------------------------------------------------|-------------------------|----------------------|-------------------------|
| РУКОВОДИЛАЦ<br>СТУДИЈСКОГ<br>ПРОГРАМА: |                         | др Милан Сечујски                              |                         |                      |                         |
| Студент:                               | Димитри                 | ie Ћук                                         |                         | Број индекса:        | EE 3/2016               |
|                                        |                         | ке науке и уграђени системи                    |                         |                      |                         |
| Ментор:                                |                         | о Марчетић                                     |                         |                      |                         |
| ·                                      |                         |                                                | ОКУМЕНТАІ               | <b>ДИЈЕ И ОДРЕДБ</b> | И СТАТУТА ФАКУЛТЕТА     |
| ИЗДАЈЕ СЕ ЗАДА                         | АТАК ЗА Д               | ИПЛОМСКИ РАД, СА СЛ                            | ІЕДЕЋИМ Е.              | ЛЕМЕНТИМА:           |                         |
|                                        | _                       | а;<br>роблема и начин практи                   | чне провере             | е резултата рада     | а, ако је таква провера |
| наслов ,                               | НАСЛОВ ДИПЛОМСКОГ РАДА: |                                                |                         |                      |                         |
| Од изворн                              | юг С кода               | до извршне бинарне с<br>микроко                | лике - комп<br>нтролера | илација и распо      | ред у меморији          |
| ТЕКСТ ЗАДАТКА:                         |                         |                                                |                         |                      |                         |
|                                        |                         |                                                |                         |                      |                         |
| Руководилац студ                       | дијског про             | грама:                                         | Ментор рад              | да:                  |                         |
| др Милан Сечујски                      |                         |                                                | др Дарко І              | <br>Марчетић         |                         |

Примерак за: о - Студента; о - Ментора

# Садржај

| 1. УВОД                                                           | 6  |
|-------------------------------------------------------------------|----|
| 1.1. Контекст и значај теме                                       | 6  |
| 1.2. Циљ рада                                                     | 6  |
| 1.3. Методологија                                                 | 7  |
| 1.4. Објашњење термина и скраћеница                               | 7  |
| 1.4.1. Хардвер и архитектура                                      | 7  |
| 1.4.2. Основни појмови софтверског система                        | 8  |
| 1.4.3. Софтверски слојеви и библиотеке                            | 8  |
| 1.4.4. Компилациони алати и процеси                               | 8  |
| 1.4.5. Формати излазних датотека                                  | 9  |
| 1.5. Релевантност теме                                            | 9  |
| 2. Улога С језика у програмирању микроконтролера                  | 11 |
| 2.1. Историјска еволуција С језика у embedded окружењима          |    |
| 2.2. Кључне особине C језика у embedded контексту                 |    |
| 2.3. Упоредна анализа С језика и алтернативних језика             |    |
| 3. Организација С изворног кода за embedded окружења              |    |
| 3.1. Основне компоненте embedded пројекта                         |    |
| 3.1.1. main.c (улазна тачка програма)                             |    |
| 3.1.2. Модули и драјвери (.с/.h парови)                           |    |
| 3.1.3. startup.s (векторска табела, <i>Reset</i> рутина, итд.)    |    |
| 3.1.4. linker.ld (меморијско мапирање)                            |    |
| 3.1.5. system_*.c (иницијализација такта, PLL, напајања)          |    |
| 3.1.6. Makefile (компилација и линковање)                         |    |
| 3.2. Употреба CMSIS и HAL слојева                                 |    |
| 3.2.1. CMSIS (Cortex Microcontroller Software Interface Standard) |    |
| 3.2.2. HAL (Hardware Abstraction Layer)                           |    |
| 3.3. Стил програмирања и стандардизација                          |    |
| 3.3.1. Индустријски стандарди кодирања за безбедност и поузданост | 26 |
| 3.3.2. Конзистентност стила и одрживост кода                      |    |
| 3.4. Приступ меморијски мапираним регистрима                      | 28 |
| 3.4.1. Меморијски мапирани улази и излази у микроконтролерима     | 28 |
| 3.4.2. Адресни простор и распоред периферија                      | 28 |
| 3.4.3. Приступ регистрима у програму (С језик)                    |    |
| 3.4.4. Моделирање хардверских регистра у С                        | 29 |
| 3.4.5. Предности и значај апстракције регистра                    | 30 |
| 4. Фазе компилације (превођења) у GCC                             | 31 |
| 4.1. Препроцесирање                                               |    |
| 4.2. Компилација                                                  |    |
| 4.3. Асемблирање                                                  |    |
| 4.4. Линковање                                                    |    |
| 4.5. Интеграција фаза компилације у оквиру GCC алатке             | 36 |
| 5. Формати резултујућих датотека                                  |    |
| 5.1. ELF формат                                                   |    |
| 5.2. Intel HEX формат                                             |    |
|                                                                   |    |

| 5.3. RAW бинарни формат (.bin)                               | 40 |
|--------------------------------------------------------------|----|
| 5.4. Motorola S-Record формат (S19)                          |    |
| 6. Употреба GNU алата: objcopy, readelf, nm, size, objdump   | 42 |
| 6.1. objcopy                                                 | 43 |
| 6.2. readelf                                                 | 44 |
| 6.3. nm                                                      | 45 |
| 6.4. size                                                    | 46 |
| 6.5. objdump                                                 |    |
| 6.6. Резиме                                                  |    |
| 7. Линкерска скрипта: MEMORY и SECTIONS дефиниције           |    |
| 7.1. MEMORY дефиниција                                       | 50 |
| 7.2. SECTIONS расподела                                      |    |
| 8. Компилација и меморијски распоред за Infineon TRAVEO T2G. |    |
| 9. Закључак                                                  | 55 |
|                                                              |    |

#### 1. Увод

# 1.1. Контекст и значај теме

Микроконтролери као уграђени рачунарски системи данас су свеприсутни у аутомобилској индустрији, индустрији аутоматике, и IoT (Internet of Things) системима. Од аутоматизованих производних погона до "паметних" уређаја и сензорских мрежа, ови мали контролери управљају широким спектром функција у реалном времену. Процењује се да је број таквих уређаја и њихова комплексност у сталном порасту, што чини знање о развоју програма за микроконтролере све важнијим. Посебан акценат је на **bare-metal** програмском моделу — односно програмирању без оперативног система — које омогућава директну контролу хардвера ради постизања максималне ефикасности. Овакво програмирање блиско хардверу сматра се кључном вештином јер пружа увид у рад система на најнижем нивоу и омогућава оптимизацију перформанси и потрошње енергије, без сувишног трошења ресурса. У контексту све веће примене уграђених система, дубље разумевање процеса развоја софтвера за те уређаје је од суштинске важности за пројектовање поузданих и безбедних система.

Језик С се издваја као доминантан у области програмирања микроконтролера захваљујући својој ефикасности и близини хардверу. Овај језик генерише машински код минималног оверхеда, који се извршава готово једнако брзо као ручно писани асемблер, што је од пресудне важности за задатке који се извршавају у реалном времену. Истовремено, С је знатно читљивији и одрживији од чистог асемблера, што олакшава тимски рад на развоју фирмвера (firmware). Због тих својстава, С (заједно са својим надскупом С++) постао је стандард у развоју bare-metal софтвера — према проценама индустрије, око 80% свих уграђених система данас користи управо С језик. Његова преносивост и стандардизација (нпр. кроз ISO/IEC С стандарде и ARM-ову CMSIS спецификацију) додатно су учврстили улогу С језика у етвеdded домену. Стога, савремени инжењери уграђених система морају добро познавати не само сам језик, већ и читав процес превођења С кода у извршни облик прилагођен мети (таргету) — хардверској архитектури микроконтролера.

# 1.2. Циљ рада

Циљ овог рада је да се систематски прикаже цео ток развоја embedded софтвера — од писања изворног кода на С језику до добијања финалне извршне бинарне слике која се уписује у меморију микроконтролера. Конкретно, рад обухвата детаљан приказ компилационог ланца уз коришћење **GCC** алатке (GNU компајлера) и анализу меморијског распореда кроз подешавање линкерске скрипте. Фокус је на томе како се изворни С код преводи (преко фазе препроцесирања, компилације и асемблирања) у објектне датотеке, затим линковањем обликује у извршну **ELF** датотеку, која се најзад конвертује у *hex* или *bin* формат погодан за учитавање у микроконтролер. Рад настоји да прикаже међусобну повезаност свих корака — од нивоа изворног кода до коначне бинарне слике — наглашавајући улогу сваког елемента у ланцу алата.

#### 1.3. Методологија

Приступ истраживању и излагању је дескриптивно-аналитички, ослоњен превасходно на званичну документацију и стандарде. Користе се референтни извори као што су техничка упутства компаније ARM (за архитектуру процесора и стандарде попут ARM Cortex-M архитектуре), документација самог GCC компилационог окружења и пратећих GNU алата, одговарајући IEEE/ISO стандарди (нпр. стандарди програмског језика С и бинарних формата), као и технички подаци произвођача микроконтролера (Infineon) укључујући технички лист (datasheet) и референтни приручник (Technical Reference Manual – TRM) за конкретни модел чипа. Анализа је спроведена кроз праћење конкретног пример пројекта и теоријско објашњење сваке фазе превођења (компилације) и распореда у меморији. Теоријски садржај је периодично илустрован практичним примером из пројекта за микроконтролер CYT2BL5CAS (породица TRAVEOTM T2G-B-E), развијеног у оквиру ModusToolboxTM окружења уз коришћење одговарајућег Board Support Package-a (BSP) KIT\_T2G-B-E\_LITE. Овај пројекат је преузет из званичног "Hello World" темплејта и служи да демонстрира стварну имплементацију концепата обрађених у раду, чиме се обезбеђује спој између теоријских појмова и практичне реализације на циљном хардверу.

# 1.4. Објашњење термина и скраћеница

#### 1.4.1. Хардвер и архитектура

- **ARM** Првобитно акроним за *Advanced RISC Machines*, данас представља и назив компаније *Arm Ltd*. и породице RISC процесорских архитектура. ARM архитектуре, нарочито линија Cortex-M, доминирају у свету микроконтролера због повољног односа између перформанси, потрошње енергије и цене. Примену налазе у мобилним уређајима, аутомобилским системима, индустријској аутоматици и ІоТ уређајима.
- **TRAVEO**<sup>TM</sup> **T2G** Породица 32-битних микроконтролера компаније *Infineon*, заснована на ARM Cortex архитектури. Другу генерацију (Traveo II) карактеришу високе перформансе, сигурносне карактеристике, као и богата периферијска подршка за аутомобилске примене. Обухвата више потпородица попут Body High и Body Entry.
- **CYT2BL5CAS** Конкретан модел микроконтролера из TRAVEO™ T2G Body Entry серије. Поседује два процесорска језгра (Cortex-M4 до 160 MHz и Cortex-M0+ до 100 MHz), 4 MB Flash меморије и велики број интегрисаних периферија. Представља циљни хардвер у оквиру овог рада.
- **Адресни простор** Целокупан скуп адреса које микроконтролер може да адресира. У Cortex-M архитектури адресни простор је линеаран и обједињује програмски код, RAM, регистре периферија и системске регистре.
- **Периферије** Хардверске компоненте интегрисане у микроконтролер које омогућавају комуникацију, управљање, мерење и генерисање сигнала (нпр. UART, ADC, PWM, GPIO, Timers). Контролишу се преко меморијски мапираних регистара.

• **Регистри** — Меморијске јединице фиксне дужине којима се управља хардвером. Разликују се системски регистри (нпр. SCB->VTOR) и регистри специфични за периферије. Често се користе у low-level програмирању и у иницијализацији система.

#### 1.4.2. Основни појмови софтверског система

- Фирмвер (firmware) Специфичан тип софтвера који се трајно налази у Flash меморији и блиско је повезан са конкретним хардвером. Задужен је за управљање основним функцијама уређаја и често не зависи од присуства оперативног система.
- bootloader Иницијални програм који се извршава по укључењу уређаја. Налази се у ROM/Flash меморији и служи за учитавање и покретање апликативног фирмвера, као и за могућност надоградње (нпр. преко UART, CAN, OTA). Може да врши криптографску валидацију садржаја.
- Дигитални потпис Криптографски механизам који обезбеђује аутентичност и интегритет бинарног кода. Најчешће се заснива на алгоритмима јавног кључа (нпр. RSA, ECDSA) и користи се у secure boot механизмима ради спречавања извршавања неауторизованог фирмвера.

#### 1.4.3. Софтверски слојеви и библиотеке

- CMSIS (Cortex Microcontroller Software Interface Standard) Званични ARM стандард који дефинише API, заглавља и структуре за приступ системским и периферним регистрима. Омогућава униформно програмирање Cortex-M уређаја независно од произвођача.
- HAL (Hardware Abstraction Layer) Софтверски слој који апстрахује приступ периферијама и хардверским ресурсима, пружајући унификоване API функције. HAL имплементације се често базирају на CMSIS-у и испоручују се као део SDK-а произвођача.
- Векторска табела Структура смештена на почетку меморијског простора (обично у Flash), која садржи адресе прекидних рутина. Први унос указује на почетак стека, други на функцију Reset Handler, док следећи одговарају одређеним прекидима.
- **Reset рутина** Почетна функција (Reset\_Handler) која се извршава по ресетовању уређаја. Задужена је за иницијализацију меморијских секција, системских параметара и покретање главне апликације (main). Представља део startup кода.

#### 1.4.4. Компилациони алати и процеси

- GCC Скуп компајлера отвореног кода (GNU Compiler Collection), који подржава различите језике (C, C++, Fortran итд.) и хардверске архитектуре. У embedded контексту, користи се GCC ARM toolchain за генерисање машинског кода за микроконтролере.
- **GNU** Пројекат слободног софтвера чији је циљ био развој потпуно отвореног оперативног система. Изнедрио је многе алате попут GCC-a, GNU binutils-a и make-a, који чине окосницу embedded build система.

• GNU binutils — Колекција алата за рад са бинарним датотекама: as (асемблер), ld (линкер), objcopy, objdump, nm, readelf итд. Користе се у фазама компилације, линковања и анализа извршних датотека.

- **Препроцесирање** Прва фаза компилационог процеса. Обрађује директиве као што су #include, #define, #ifdef, проширује макрое и формира јединствен .i фајл са чистим С кодом без препроцесорских ознака.
- **Компилација** Претварање .i фајла у асемблерски (.s). Анализира се синтакса и семантика С кода, врши се типска провера и генерише се машински независан асемблерски код.
- **Асемблирање** Преводи .s у објектни бинарни фајл (.o). У овој фази се генеришу машинске инструкције, секције, табеле симбола и relocation уноси.
- Мапирање меморије / Линковање Процес који спаја више .o и библиотека у једну .elf датотеку. Линкер, по инструкцијама из скрипте, смешта секције (.text, .data, .bss итд.) у одговарајуће адресне просторе.
- **Линкерска скрипта** Текстуална датотека која описује распоред и величине меморијских региона (MEMORY {}) и секција (SECTIONS {}). Одређује како ће објекатни код бити постављен у Flash и RAM уређаја.

#### 1.4.5. Формати излазних датотека

- **ELF** *Executable and Linkable Format* је стандардни бинарни формат за Unix-подобне системе. ELF фајл садржи све секције програма, симболе и релокационе информације, што га чини погодним за дебаг и анализу.
- **Intel HEX формат** ASCII формат у којем се бинарни подаци представљају као хексадекадне вредности по линијама (record-има). Сваки запис садржи адресу, тип података, сам садржај и контролни збир. Широко је подржан од стране програматора и дебагера.
- **Бинарни формат** "RAW" представљање садржаја меморије, без заглавља и симболичких информација. .bin фајлови садрже само податке предвиђене за Flash и погодни су за директно програмирање.
- **Motorola S-Record формат** Алтернативни текстуални формат сличан Intel HEX-у. Користи S0—S9 записе који садрже адресу, дужину, податке и контролни збир. Широко подржан у индустријским програматорима и legacy алатима.

# 1.5. Релевантност теме

Разматрана тематика директно одговара потребама реалног развоја софтвера у уграђеним окружењима, нарочито за системе који раде без оперативног система (baremetal). Правилна организација изворног кода, разумевање формата резултујућих датотека и контрола над меморијским распоредом представљају предуслов за функционалан и безбедан рад микроконтролера у критичним применама – попут

аутомобилских управљачких система, медицинских уређаја или индустријских контролера. Незнање или превид у овим областима може довести до тешко уочљивих грешака које компромитују поузданост система. Са друге стране, темељно разумевање компилационог процеса и меморијске структуре фирмвера отвара пут ка напреднијим темама. Конкретно, знања изложена у овом раду представљају основу за имплементацију **bootloader-**а (који захтева прецизно управљање меморијским адресним просторима и секцијама кода), за увођење механизама безбедности фирмвера (нпр. верификацију кода путем дигиталног потписа), као и за оптимизацију потрошње ограничених ресурса микроконтролера. На тај начин, ова тема је значајна не само теоријски, већ и практично – као неопходан део знања за инжењере који се баве развојем поузданог и сигурног embedded софтвера.

# 2. Улога С језика у програмирању микроконтролера

Језик С је најзаступљенији у програмирању микроконтролера због своје флексибилности и ефикасности (мали оверхед). Под флексибилношћу се подразумева могућност директног приступа хардверским ресурсима, као што су меморијске адресе и регистри. Са друге стране, мали оверхед значи да компајлирани код заузима минимално меморије и омогућава брзо извршавање. Захваљујући овим особинама, С омогућава оптимално коришћење ограничених ресурса карактеристичних за уграђене системе.

Практично све – од малих контролера у уређајима до оперативних система – може бити написано у С-у због његове преносивости и способности да уз минималне наредбе пружи максималну контролу над хардвером. У домену уграђених система, С пружа низак ниво апстракције: омогућава директан приступ меморијским адресама и периферијама, управљање битовима и регистрима, као и прецизну контролу над временски критичним секцијама кода. За разлику од виших језика, компајлер С језика генерише ефикасан машински код који се извршава готово једнако брзо као ручно писани асемблер, чиме је погодан за примене у реалном времену (real-time). Истовремено, С код је знатно читљивији и одрживији од чистог асемблера, што је важно при тимском развоју софтвера. Захваљујући овим особинама, С (и његов надскуп С++) је постао стандард у развоју фирмвера за микроконтролере, омогућивши и преносивост кода између различитих архитектура. Другим речима, С обезбеђује директан приступ хардверу и високе перформансе, што су кључни захтеви у embedded систему.

# 2.1. Историјска еволуција С језика у embedded окружењима

Језик С је развијен раних 1970-их у Bell Labs-у за потребе оперативног система UNIX, али је због своје минималистичке и ефикасне структуре веома брзо нашао примену у embedded системима. Током развоја микроконтролера и појаве 8-битних и 16-битних архитектура, С је потиснуо асемблер као доминантни језик због боље читљивости и лакше преносивости. Данас, С представља основну полазну тачку за развој софтвера у реалновременским (real-time) и ресурсно ограниченим системима, посебно захваљујући стандардизацији преко ISO/IEC 9899:2018 (C18) и спецификацијама као што је CMSIS (Cortex Microcontroller Software Interface Standard) од стране ARM-а.

Ознака ISO/IEC 9899:2018 (С18) односи се на званични међународни стандард за програмски језик С, који су усвојиле две признате организације за стандардизацију: International Organization for Standardization (ISO) и International Electrotechnical Commission (IEC). Означење 9899 представља број стандарда који се односи на језик С, док 2018 указује на годину последње ревизије. Ознака С18 је неформална, али широко прихваћена у стручној заједници, и односи се на ову верзију С стандарда. Стандард С18 представља мању ревизију претходне верзије С11 (ISO/IEC 9899:2011), фокусирајући се на техничке исправке, унапређење формалне прецизности и уклањање неусаглашености, без увођења нових синтаксних или семантичких елемената. Због тога се С18 сматра најстабилнијом и најсавременијом верзијом стандарда језика С која се тренутно примењује у индустријским и академским окружењима.

# 2.2. Кључне особине С језика у embedded контексту

У контексту програмирања микроконтролера, С се издваја следећим особинама:

- Директан рад са меморијом Показивачи омогућавају манипулацију на нивоу бајта и регистра, што је неопходно за рад са периферијама. Кроз volatile квалификатор могуће је обезбедити исправно понашање при асинхроним изменама вредности (нпр. од стране хардвера или прекида).
- Фино управљање меморијским распоредом Коришћењем \_\_attribute\_\_((section(".example"))), програмер може утицати на то у коју меморијску регију ће одређена функција или променљива бити смештена, што је критично у систему без оперативног система.
- **Интеграција са асемблером** Када је неопходна максимална оптимизација или директна манипулација регистрима, С омогућава уметање асемблерских инструкција (\_asm\_\_) или позиве на екстерне рутине, што га чини флексибилним алатом за развој близак хардверу (low-level programming).
- **Прецизна контрола времена извршавања** У real-time системима детерминистичност је од пресудне важности. С не садржи runtime механизме попут garbage collector-a, чиме обезбеђује временски предвидљиво понашање.

# 2.3. Упоредна анализа С језика и алтернативних језика

Иако се у embedded индустрији појављују и други језици (нпр. Rust, Ada, Python/MicroPython), ниједан не достиже ниво подршке и зрелости који има С. Python се користи углавном у образовне и прототипске сврхе, Rust још увек има ограничену подршку за специфичне архитектуре и toolchain-ове, док је Ada присутна углавном у високо-регулисаним индустријама (авионика, нуклеарна техника).

С се тако намеће као оптималан компромис између перформанси, поузданости, контроле и одрживости. Он пружа довољну блискост хардверу за временски критичне апликације, а истовремено омогућава тимски развој, проверу стандарда као што су MISRA и лако повезивање са библиотекама и хардверским апстракционим слојевима.

# 3. Организација С изворног кода за embedded окружења

Развој софтвера за микроконтролере у програмском језику С захтева дисциплиновану и модуларну организацију изворног кода, посебно у условима где не постоји оперативни систем и где је програм одговоран за директно управљање хардвером — такозвано baremetal програмирање. У овом контексту, коректна организација пројекта није само питање структуралне естетике, већ предуслов за исправно иницијализовање система, ефикасну меморијску расподелу и могућност проширивости и тестирања.

## 3.1. Основне компоненте embedded пројекта

Типичан С-пројекат за ARM Cortex-M4 микроконтролер у bare-metal окружењу састоји се из више међусобно повезаних компоненти. Произвођач микроконтролера обично испоручује почетни startup код (нпр. асемблерску или С датотеку) са векторском табелом прекида и reset рутином, као и одговарајућа CMSIS заглавља и HAL библиотеке за рад са периферијама. На основу тога, програмер развија сопствени изворни код у main.c датотеци и повезаним модулима (нпр. сензори, управљање мотором), укључујући потребне хедере за сваки модул. Поред тога, пројекат садржи и линкерску скрипту (нпр. linker.ld) која описује меморијске регије микроконтролера (Flash, RAM) и дефинише распоред секција програма (.text, .data, .bss, stack итд.) у те регије. Оваква структура обезбеђује да сваки део кода и података буде смештен на предвиђено место током линковања, што је основа за стабилан и поуздан embedded систем.

#### 3.1.1. main.c (улазна тачка програма)

**main.c** је главна изворна јединица програма која садржи функцију main(), односно улазну тачку извршавања. У овој датотеци врши се иницијализација хардвера и система по покретању микроконтролера, након чега програм прелази у главну петљу (тзв. *superloop*) у оквиру које се обавља његова главна функционалност. Типично се у main() функцији омогућавају прекиди и покрећу иницијализације свих потребних периферија, па затим следи бесконачна петља која одржава рад програма. Уколико систем користи RTOS, у main() се уместо бесконачне петље може стартовати *scheduler*, али у *bare-metal* приступу main() сам управља током извршавања.

У наставку је приказан један могући поједностављен пример структуре функције main(). После ресета, најпре се иницијализују плоча и периферије позивом функције за подешавање хардвера (у овом случају cybsp\_init()), а резултат иницијализације се проверава. Потом се омогућавају глобални прекиди, након чега би следила подешавања комуникације (нпр. UART за debug излаз) и осталих уређаја, па улазак у главну петљу програма:

```
int main(void)
{
    result = cybsp_init(); /* Иницијализација платформе и периферија */

    if (result != CY_RSLT_SUCCESS)
    {
        CY_ASSERT(0); /* Заустави извршење ако иницијализација није успела */
    }

    __enable_irq(); /* Омогућавање глобалних прекида */

    /* Укључење UART-а и LED */
    cy_retarget_io_init_fc(...);
    cyhal_gpio_init(CYBSP_USER_LED, ...);
```

```
timer_init(); /* Старт тајмера који генерише прекид сваке секунде */
while (true)
{
    if (timer_interrupt_flag)
    {
        timer_interrupt_flag = false;
        cyhal_gpio_toggle(CYBSP_USER_LED); // трептање LED-ом
    }
}
```

Горњи пример демонстрира типичне кораке на почетку main() функције – иницијализацију хардвера и омогућавање прекида. Након тога, main() обично улази у бесконачну петљу (while(1) или for(;;)) у којој се обрађују догађаји или сензорски подаци, шаљу поруке, управља актуаторима и сл. (нпр. очитавање УАРТ улаза и укључивање/искључивање LED диоде у примеру). Напомена: конкретна реализација main() може знатно да варира у зависности од пројекта – наведени код је само једна могућа варијанта имплементације.

#### 3.1.2. Модули и драјвери (.c/.h парови)

Већи пројекти се организују на модуларан начин, тако да се поједине функционалне целине реализују у виду одвојених модула (нпр. gpio.c, uart.c, sensor.c, motor\_control.c, итд.). Сваки такав модул обично долази у пару: изворна датотека са имплементацијом (.c) и одговарајућа заглавна датотека (.h) која декларише његов јавни интерфејс. Заглавни (.h) фајл садржи прототипове функција, декларације структура, епит типова и глобалних променљивих које модул излаже другим деловима програма. На тај начин се постиже јасна подела кода и боља могућност поновне употребе и тестирања — остале јединице укључују само потребне хедере и позивају функције модула преко дефинисаног интерфејса.

Пример једног таквог модула је драјвер за тајмер. У његовом заглављу може бити декларација функције за покретање тајмера, на пример:

```
cy_rslt_t cyhal_timer_start(cyhal_timer_t *obj);
```

Ова функција је у .h датотеци само најављена (прототип), а у изворној .c датотеци дата је њена реализација. У коду испод видимо делић имплементације функције cyhal\_timer\_start у оквиру драјвера тајмера: након провера објекта и услова, функција позива рутине блиске хардверу за подешавање периферије — укључивање бројача и стартовање тајмера:

```
if (CY_RSLT_SUCCESS == result)
{
    result = Cy_TCPWM_Counter_Init(obj->tcpwm.base, ..., config); /* Иницијализација хардверског
тајмер блока */
}
```

```
if (CY_RSLT_SUCCESS == result)
{
    Cy_TCPWM_Counter_Enable(obj->tcpwm.base, ...); /* Enable тајмер (покретање бројања) */
}
```

У овом сегменту кода функција драјвера користи функције из произвођачке PDL библиотеке (Cy\_TCPWM\_Counter\_Init/Enable) да конфигурише и покрене одговарајући тајмерски периферни блок микроконтролера. На тај начин се остварује апстракција – виши нивои кода позивају једноставну cyhal\_timer\_start() функцију, док она интерно обавља комплексне операције над регистрима. *Напомена*: структура модула и стил имплементације могу се разликовати; приказани пример је само један могући начин организације .c/.h пара датотека у склопу драјвера.

#### 3.1.3. startup.s (векторска табела, *Reset* рутина, итд.)

асемблерску Startup датотека (често названа startup.s за или **startup.c** за  $\mathbf{C}$ имплементацију) садржи код који се извршава први након укључења или ресетовања микроконтролера. Њене главне улоге су: (1) дефинисање векторске табеле прекида, која на познатој адреси (нпр. почетак Flash меморије) садржи почетне адресе свих рутина, укључујући и почетну вредност стек показивача адресу Reset Handler-a; (2) имплементација саме Reset Handler рутине, која припрема извршно окружење пре него што се позове функција main().

**Векторска табела** је низ од 32-битних вредности које одговарају почетном стек показивачу и адресама свих излазних тачака прекида. На примеру испод видимо почетак векторске табеле за један Cortex-M4 уређај — прва вредност је иницијални адресни врх стека (**Stack Top**), а затим следе адресе обрадних рутина: Reset\_Handler, *Non-Maskable Interrupt* (NMI), *HardFault*, и осталих дефинисаних изузетака и прекида:

```
Vectors:
.long
        StackTop
                     /* Почетна адреса стека */
      Reset Handler /* Reset Handler */
.long
.long
      CY NMI HANLDER ADDR /* NMI Handler */
.long
      HardFault Handler /* Hard Fault Handler */
      MemManage Handler /* MPU Fault Handler */
.long
      BusFault Handler
                         /* Bus Fault Handler */
.long
      UsageFault Handler /* Usage Fault Handler */
.long
               /* (наставак листе прекида) */
```

Након векторске табеле, *startup* код реализује саму *Reset\_Handler* функцију. Ова рутина се извршава на самом почетку (на њу упућује други елемент векторске табеле) и њен задатак је да припреми окружење за С програм. То типично обухвата: постављање почетног стека, копирање иницијализационих података из Flash у RAM (секција .data), брисање (иницијализација на нуле) неиницијализованих статичких променљивих (секција .bss), потенцијално омогућавање FPU јединице, подешавање векторске табеле ако се преселила у RAM, и позив функције за почетно подешавање система (нпр. SystemInit()). Тек након тога, *startup* рутина позива корисничку функцију main() и предаје јој даљу контролу извршавања програма. На крају *Reset Handler*-а се обично

налази бесконачна петља као заштита ако main икада врати управљање (што се у исправном програму не дешава):

```
/* ... (иницијализација .data и .bss секција) ... */
#ifndef __NO_SYSTEM_INIT
bl SystemInit /* Позив функције за системску иницијализацију */
#endif

/* ... (позив конструктора C++ објеката) ... */
bl __libc_init_array

/* Execute main application */
bl main /* Позив корисничке main() функције */

/* Call C/C++ static destructors */
bl __libc_fini_array

/* Should never get here */
b . /* Бесконачна петља (dead loop) */
```

Горњи код илуструје завршни део *startup* секвенце: након припреме меморије, позива се SystemInit (осим ако није искључен макроом), затим рутина за статичке конструкторе (\_\_libc\_init\_array), па корисничка функција main. По повратку из main (који се у правилу не дешава у *bare-metal* програмима), позвали би се деструктори статичких објеката и програм улази у бесконачну петљу. *Напомена*: конкретан садржај *startup* кода зависи од конкретног архитектурног језгра и алатног ланца – приказани пример одговара CMSIS шаблону за Cortex-M4 и једно могуће извођење *reset* рутине.

# 3.1.4. linker.ld (меморијско мапирање)

**Линкерска скрипта или директива** (најчешће названа **linker.ld**) одређује како ће се секције кода и података распоредити у физичкој меморији микроконтролера током процеса линковања. Она описује расположиве меморијске регије (нпр. флеш и рам) и правила смештања различитих секција програма у те регије. Тиме линкер зна тачно на које адресе треба ставити сваки део извршног кода и података, што је од критичне важности у *bare-metal* систему где нема оперативног система да динамички управља меморијом.

У делу ниже видимо пример дефиниције меморијских регија у линкерској скрипти. Дефинисана су два главна региона: FLASH (са атрибутима *rx* (read execute) — за извршавање и читање) од адресе 0x10000000 дужине 0x410000 бајтова, и RAM (са rwx (read write execute) атрибутима) од адресе 0x08020000 дужине 0x5F800 бајтова. Ове адресе и величине одговарају конкретном микроконтролеру (овде пример двојезгарног система где СМ4 језгро користи одређени део меморије):

```
ram (rwx): ORIGIN = 0x08020000, LENGTH = 0x5F800 flash (rx): ORIGIN = 0x10000000, LENGTH = 0x410000 sflash_user_data (rx): ORIGIN = 0x17000800, LENGTH = 0x800 /* специјални Flash */ ...
```

Након дефинисања меморије, линкерска скрипта описује распоред секција. На пример, код за Cortex-M4 језгро може поставити секцију .text (садржи извршни код програма) у Flash на адресу одмах након резервисаног простора за M0+ језгро (ако постоји). Секција .data (иницијализовани подаци) мора бити смештена у RAM, али њене иницијалне вредности треба сачувати у Flash – што се постиже директивом AT> flash у скрипти. Извод из линкер скрипте који то илуструје:

```
.data __ram_vectors_end__ : AT>flash {
    ...
    __data_end__ = .;
} > ram
```

У горњем примеру, секција .data се алоцира у RAM (ознака > ram), али јој је *Load Memory Address* постављена на Flash (AT>flash). То значи да ће сви бајтови .data секције бити уписани у извршну датотеку на одговарајућим Flash адресама, одакле ће их *startup* код копирати у RAM при покретању. Слично, секција .bss (неиницијализовани подаци) дефинише се са атрибутом NOLOAD и смешта у RAM, чиме линкер означава да за њу не треба резервисати простор у Flash фајлу већ ће бити само обележена за касније попуњавање нулама у RAM-у. Линкерска скрипта обично додељује и симболе као што су \_\_StackTop и \_\_StackLimit на крају RAM меморије, чиме се дефинише позиција и величина стека програма.

Добро осмишљена линкерска скрипта обезбеђује исправно мапирање целокупног програма у меморију микроконтролера. *Напомена*: иако постоје унапред припремљене генеричке скрипте, увек је потребно прилагодити их конкретном чипу (према подацима из *datasheet*-а) како би се сви сегменти (нпр. више блокова RAM-а, посебне меморије) исправно обухватили.

# 3.1.5. system\_\*.c (иницијализација такта, PLL, напајања)

Уз startup код, уобичајено постоји посебна je да И датотека назива облика system <device>.c, која садржи функције за почетну конфигурацију система такта и напајања. ARM CMSIS стандард предвиђа функцију SystemInit() у овој датотеци, коју startup позива непосредно пре корисничког кода. Улога SystemInit() је да подеси основне параметре система: такт микроконтролера (нпр. учитава унутрашњи осцилатор или подешава PLL множилац и делитеље такта за жељену фреквенцију), подеси брзину рада Flash меморије (нпр. wait-state-ове) у складу са тактом, омогући FPU (уколико постоји) и припреми глобалну променљиву SystemCoreClock која садржи вредност фреквенције језгра. Ова датотека је специфична за сваки *device* и типично је испоручује произвођач – програмер је обично не мења, осим ако је потребно прилагодити такт нестандардно.

Уколико се користи произвођачки *Hardware Abstraction Layer* (HAL), део системске иницијализације може бити распоређен и у функције за иницијализацију плоче или периферија. На пример, Infineon-ова функција cybsp\_init() позива низ потпроцедура које укључују подешавање хардвер менаџера ресурса и система напајања:

Овај фрагмент кода илуструје да позивом једне функције (cybsp init) у main.c заправо покрећемо вишеструка подешавања у позадини – од менаџмента тактова и напона до резервисања ресурса за вишејезгарне системе (нпр. функције cycfg config init() и друге у наставку кода). У класичној CMSIS поставци, сличне акције обавља SystemInit(), али у HAL иницијализације специфичне OBOM примеру оне cy део произвођача. Напомена: без обзира на конкретну реализацију, суштина system \*.c јесте да се сви кључни системски параметри микроконтролера подесе на самом почетку (пре апликационог кода), како би остатак програма могао да ради на предвидљивој тактној фреквенцији и конфигурацији.

#### 3.1.6. Makefile (компилација и линковање)

**Makefile** представља скрипт за аутоматизацију процеса превођења кода и линковања у извршну бинарну слику. У GCC окружењу, *Makefile* прописује кораке: који се фајлови требају компајлирати, са којим опцијама, и како их затим повезати линкером. На пример, наредба:

```
arm-none-eabi-gcc -O2 -mcpu=cortex-m7 -o program.elf main.c uart.c startup.s -T linker.ld
```

илуструје како се у једном кораку могу обавити све фазе превођења — наведеном командом GCC ће аутоматски препроцесирати и компајлирати main.c, uart.c и асемблерски startup.s, а затим их линковати користећи линкер скрипту -Т linker.ld, производећи извршни ELF фајл (program.elf). У пракси, Makefile управо генерише овакве командне позиве за све изворне јединице пројекта, укључујући и додавање неопходних путева до заглавља, библиотека и дефинисање макроа за условну компилацију. Он такође води рачуна о редоследу извршавања — да се сваки .c преведе у .o пре линковања, да се асемблерске датотеке такође преведу, и на крају да се позове линкер са свим насталим објектним фајловима и одговарајућом .ld скриптом.

У случају интегрисаних развојних окружења (IDE) као што су IAR или Keil, не постоји експлицитан *Makefile*, али концепт је исти — пројекат садржи подешавања која дефинишу који се фајлови компајлирају и како, а IDE интерно генерише командне позиве компајлера и линкера. Било да се користи ручно написан *Makefile* или IDE, резултат је на крају исти: сви претходно описани делови пројекта (startup код, main.c, модули, системске функције и линкерска скрипта) бивају састављени и повезани у једну извршну бинарну слику спремну за учитавање у микроконтролер. *Напомена*: конкретна синтакса и организација *Makefile*-а могу бити различити (нпр. коришћење CMake уместо ручног *Makefile*-а), али увек служе истој сврси — аутоматизацији и контролисању процеса грађења *embedded* софтвера.

#### 3.2. Употреба CMSIS и HAL слојева

У развоју софтвера за микроконтролере, уобичајено је ослањање на стандардизоване слојеве апстракције (библиотеке) који поједностављују руковање хардвером. Два најважнија таква слоја су **CMSIS** и **HAL**. Они заједно обезбеђују структуриран приступ компонентама система – од самог процесорског језгра до периферијских уређаја – чиме се смањује сложеност директног руковања регистрима и убрзава развој. У наставку су описани ови слојеви и њихова улога, уз пример који илуструје њихову употребу у пракси.

# 3.2.1. CMSIS (Cortex Microcontroller Software Interface Standard)

**CMSIS** је стандард који је развио **ARM** са циљем да уједначи софтверски интерфејс за Cortex-M микроконтролере. CMSIS обезбеђује дефиниције и функције блиске хардверу за сам процесор и основне периферије, независно од произвођача конкретног чипа. Кроз CMSIS, произвођачи микроконтролера испоручују сет заглавља и рутина које описују хардвер на симболичком нивоу — од регистара језгра до специјализованих периферијских јединица — на стандардизован начин.

Конкретно, CMSIS укључује **CMSIS-Core** део, који обухвата дефиниције за све регистре процесорског језгра и основне периферије. На пример, заглавља попут *core\_cm4.h* или *core\_cm7.h* садрже структуре и адресне симболе за Cortex-M4/M7 регистре (попут регистра за векторски адресер прекида **SCB->VTOR**), док заглавље *system\_<i>Device</i>h* (нпр. *system\_stm32f4xx.h* за STM32 или одговарајуће Infineon заглавље за Traveo T2G) садржи параметре такта и почетну функцију за подешавање система. Захваљујући тим дефиницијама, програмер може да приступа регистрима на читљив начин уместо кроз "магичне" бројеве адреса — на пример, да упише вредност у регистар контролера прекида употребом симбола **NVIC->ISER** уместо ручног адресирања меморије.

Важно је нагласити да CMSIS пружа и стандардизовану шаблон-рутину за стартап (startup) система. При укључивању микроконтролера, извршава се *startup* код (написан у асемблеру или С-у) који долази уз CMSIS пакет за тај уређај. Овај почетни код дефинише **векторску табелу прекида** (листа адреса свих прекидних рутина) и садржи *Reset\_Handler* функцију (рутину на коју процесор прелази након ресетовања). У оквиру *Reset\_Handler*-а се обично иницијализују основне ствари: пуњење почетних вредности података у RAM, брисање *BSS* секције, конфигурисање такта система и осталих низводних компоненти. Према CMSIS стандарду, уобичајено је да се у склопу стартуп кода позове функција **SystemInit()** – дефинисана у *system\_* заглављу – која подешава системски часовник (такт) и друге основне параметре пре него што се настави ка функцији тајп. На тај начин, CMSIS обезбеђује да сваки микроконтролер има предвидљиво иницијално окружење за извршавање корисничког кода, без оперативног система.

Кроз овакав слој, програмер има конзистентан и типски безбедан начин да управља хардвером. На пример, ако је потребно померање векторске таблице (нпр. при коришћењу bootloader-a), довољно је подесити регистар **SCB->VTOR** на адресу нове таблице – CMSIS је већ обезбедио симболички назив *SCB* (System Control Block)

структуре и поље *VTOR*. Слично томе, омогућавање или забрана прекида врши се стандардизованим функцијама попут <u>enable\_irq()</u> и <u>disable\_irq()</u>, које су имплементиране као инлајн асемблерске инструкције у CMSIS заглављима. Ове функције раде униформно без обзира на конкретан компајлер, што доприноси преносивости кода.

СМЅІЅ самим својим постојањем смањује могућност грешака и повећава читљивост кода. Захваљујући њему и одговарајућим заглављима које обезбеђује произвођач, инжењери више не морају да користе непрегледне изразе за директан упис у меморију (нпр. \*(volatile uint32\_t\*)0x50000004 = 0x1;), већ могу да користе симболичке и читљиве облике као што је GPІО->ОUТ = 0x1; за управљање излазима – што значајно унапређује одрживост и транспарентност софтвера. При томе, CMЅІЅ не додаје практично никакав оверхед при превођењу: коришћење CMЅІЅ макроа и структура своди се на директне операције над регистрима, па су перформансе таквог кода једнаке као да су регистри адресирани ручно. Ова особина чини CMЅІЅ погодним и за критичне делове система где је битна брзина извршавања и детерминистичко понашање.

#### 3.2.2. HAL (Hardware Abstraction Layer)

**HAL** представља слој **апстракције хардвера** који углавном обезбеђује произвођач микроконтролера у виду библиотека. За разлику од CMSIS-а, који је оријентисан на сам процесор и основне регистре, HAL библиотеке циљају више нивое — пружају готове функције за управљање разним периферијама (тајмерима, УАРТ-ом, GPIO линијама, А/D конверторима итд.), скривајући детаље реализације. Идеја HAL-а је да понуди униформан интерфејс за честе операције, тако да програмер може, на пример, једноставном функцијом да пошаље податке преко серијског порта или генерише PWM сигнал, без потребе да познаје сваки бит у неколико регистара тог периферног модула.

Типичан пример је STMicroelectronics-ова HAL библиотека за STM32 серију контролера: она нуди функције као што је HAL\_UART\_Transmit() за слање података преко UART-а или HAL\_GPIO\_WritePin() за подешавање излаза на пину. Слично томе, Infineon (раније Cypress) за своје PSoC/Traveo микроконтролере пружа HAL функције попут cyhal\_gpio\_init() за конфигурацију пина или cyhal\_timer\_start() за управљање тајмером. Ове функције унутар себе обављају читав низ корака — од укључивања такта периферије, конфигурисања режима рада, до провере валидности параметара — али су ка кориснику изложене као једноставан API (Application Programming Interface). На тај начин, **HAL слојеви смањују количину хардверски зависног кода у главном програму**: велики део посла обављају HAL рутине, а код програмера постаје краћи и јаснији.

Важно је напоменути да HAL библиотеке пишу сами произвођачи за своје породице уређаја, па оне нису универзално преносиве између различитих брендова микроконтролера. Међутим, унутар једне породице или једног произвођача, HAL настоји да уједначи интерфејсе. То значи да прелазак са једног модела на други (нпр. између различитих STM32 чипова или између различитих Infineon PSoC модела) захтева минималне измене у коду ако се користи HAL. Библиотека апстракује разлике: сваки конкретан модел ће имати другачије регистре "испод хаубе", али ће позив функције HAL\_UART\_Transmit() радити на свима њима на исти начин са аспекта програмера.

Овакав приступ убрзава развој и чини прототипе брже готовим, јер се инжењер може фокусирати на логику програма уместо на детаље иницијализације сваког подсистема.

Цена те погодности је одређени **оверхед** – у погледу меморије и брзине. НАL функције су општије и садрже додатне провере и слојеве позива, па генерисани код може бити спорији у односу на ручно оптимизовани приступ регистрима. Ипак, у већини случајева овај оверхед је прихватљив, поготово имајући у виду добитак у преносивости и уштеди времена приликом софтверског развоја. За **перформансно критичне секције**, добра пракса је да се НАL користи за већи део система, а да се само у уским грлима где је потребна максимална брзина прибегне директном приступу регистрима (коришћењем CMSIS симбола или специјализованих *Low Level* драјвера). На тај начин се постиже баланс између брзине и одрживости кода.

**Пример употребе HAL и CMSIS:** Размотримо једноставан програм који трепће диодом на развојној плочи са Infineon Traveo II Т2G микроконтролером. Захваљујући HAL слоју, целокупна иницијализација хардвера и периферија своди се на неколико позива функција из библиотеке, док CMSIS обезбеђује основне операције на нивоу језгра. У наставку је извод из функције та птаквог програма (поједностављено за приказ):

```
int main(void)
{
   cy_rslt_t result;
   result = cybsp_init();
                             /* Иницијализација хардвера платформе и периферија */
   if (result != CY_RSLT_SUCCESS)
   {
       CY_ASSERT(0);
                            /* Заустави извршавање ако иницијализација није успела */
   }
    __enable_irq();
                             /* Омогућавање глобалних прекида */
   /* Иницијализација корисничке LED диоде (GPIO пина) */
    cyhal gpio init(CYBSP USER LED, CYHAL GPIO DIR OUTPUT,
                   CYHAL GPIO DRIVE STRONG, CYBSP LED STATE OFF);
    /* Покретање тајмера који ће генерисати прекид сваке секунде */
    cyhal_timer_t led_blink_timer;
    cyhal_timer_init(&led_blink_timer, NC, NULL);
   cyhal_timer_set_frequency(&led_blink_timer, 10000);
                                                            // подеси извор такта
   cyhal_timer_configure(&led_blink_timer, &led_blink_timer_cfg); // конфигурација периода
   cyhal_timer_register_callback(&led_blink_timer, isr_timer, NULL);
    cyhal_timer_enable_event(&led_blink_timer, CYHAL_TIMER_IRQ_TERMINAL_COUNT, 7, true);
    cyhal_timer_start(&led_blink_timer);
   for(;;)
        /* У главној петљи проверава се флаг који поставља тајмерски прекид */
        if (timer_interrupt_flag)
        {
            timer interrupt flag = false;
```

```
cyhal_gpio_toggle(CYBSP_USER_LED); // трептање LED-ом - инверзија стања пина
}
}
```

Горњи код илуструје како се **HAL** функционалност користи на високом нивоу, док су детаљи скривени у позадини. На пример, позив cybsp\_init() иницира читав низ операција неопходних да микроконтролер правилно проради: подешава се системски такт, покрећу се модули за управљање напајањем, резервишу се хардверски ресурси и иницијализују подразумеване периферије на плочи. Све те активности се одвијају "испод хаубе" у оквиру неколико функција које ова рутина позива. У конкретном случају Infineon Traveo T2G платформе, cybsp\_init() интерно позива, између осталог, функцију за покретање менаџера хардверских ресурса (**cyhal\_hwmgr\_init(**)) и функцију за подешавање система напајања (**cyhal\_syspm\_init(**)). Такође се примењују унапред генерисана подешавања такта и пинова (у оквиру функција као што су cycfg\_config\_init() и сусfg\_config\_reservations()), и региструју се повратни позиви за промену такта при уласку микроконтролера у режим ниске потрошње. Све ово је спаковано у једно апстрактно **HAL** позивно место, тако да у та функцији имамо само један ред којим "магично" спремамо читав систем за рад. Овај приступ очигледно поједностављује структуру програма и смањује могућност пропуста у иницијализацији.

Након успешне иницијализације, у главној рутини се позива CMSIS функција enable irq() да се омогуће глобални прекиди на нивоу процесора. Ово је неопходан корак који је илустрација сарадње између CMSIS-а и HAL-а: CMSIS брине о контролеру прекида (NVIC) и другим системским аспектима, док HAL преузима конфигурисање периферијских модула који ће те прекиде користити. У примеру, након што је тајмер конфигурисан и стартован позивима cyhal\_timer\_\* функција (HAL апстракција за тајмерски блок), тајмерски хардвер аутономно броји време и генерише прекид сваке секунде. Тај прекид се обрађује у позадини (HAL је регистровао isr timer обрађивач преко cyhal timer register callback), при чему та обрада постави заставицу timer interrupt flag. Главна петља програма (for(;;)) затим, уз помоћ тог флага, зна када је једна секунда протекла и у одговору позива cyhal gpio toggle() – HAL функцију која мења стање излазног пина где је прикључена LED диода. Резултат је трептање диоде у интервалу од 1Hz, остварено без иједног директног уписа у хардверски регистар у корисничком коду. Сви уписи (нпр. подешавање излаза пина или конфигурација тајмера) реализовани су унутар HAL функција, користећи при том CMSIS дефинисане симболе за приступ одговарајућим регистрима.

Овај пример показује предности слојевитог приступа. Код је знатно читљивији и краћи него што би био уколико бисмо ручно конфигурисали сваки регистар. Истовремено, захваљујући CMSIS-у, имамо сигурност да су системски ресурси (попут векторске таблице, стања прекида, почетних секција меморије) исправно постављени пре него што НАL крене са иницијализацијом периферија. CMSIS обезбеђује формалну доследност и стабилну основу система, док HAL омогућава бржи развој и већу преносивост кода између различитих контролера исте породице. Комбинацијом ова два слоја, програмер може да достигне оптималан спој поузданости и ефикасности: критични делови се по потреби могу писати ближе хардверу (користећи CMSIS директно за приступ регистрима), док се већи део апликације ослања на проверене HAL рутине које убрзавају израду и смањују могућност грешака. Таква систематична организација кода

у слојевима значајно олакшава разумевање целокупног система и одржавање програма током његовог животног циклуса.

# 3.3. Стил програмирања и стандардизација

Развој С кода за уграђене (embedded) системе мора да следи дисциплинован стил и јасно дефинисане стандарде због високе поузданости и дугог животног циклуса који се од ових система очекују. Посебно у критичним доменима (аутомобилска индустрија, индустријска аутоматика, медицински уређаји), софтверски инжењери примењују строге смернице програмирања како би смањили могућност грешака и неодређеног понашања програма. Ове смернице обухватају како општи стил кодирања (конзистентно форматирање, именовање и организацију кода), тако и формалне стандарде безбедног програмирања усмерене на спречавање грешака на нивоу језика.

#### 3.3.1. Индустријски стандарди кодирања за безбедност и поузданост

Најзначајнији скуп правила за стил и безбедност кода у индустрији уграђених система је MISRA С стандард (Motor Industry Software Reliability Association). MISRA С дефинише строга правила којих се програмери требају придржавати како би избегли неодређено или потенцијално опасно понашање програма. Ова правила, између осталог, укључују забрану коришћења динамичке алокације меморије (нпр. функција malloc), неконтролисаних конверзија типова (cast операција) и употребе goto наредби. Придржавање оваквих стандарда омогућава примену формалне верификације и аутоматизоване статичке анализе кода (помођу алата као што су PC-lint или Coverity), чиме се значајно повећава поузданост и безбедност резултујућег софтвера. У домену аутомотива, поштовање MISRA смерница је де-факто обавезно за испуњавање захтева функционалне безбедности (нпр. у оквиру стандарда ISO 26262 за аутомобилске системе).

Поред MISRA-е, постоје и други сетови смерница усмерени на побољшање квалитета и сигурности кода. Један од њих је CERT C стандард, који представља смернице за сигурно програмирање на C језику. Док је MISRA првенствено фокусиран на безбедност система и избегавање кварова (safety) у уграђеним уређајима, CERT C нагласак ставља на обезбеђивање софтвера од рањивости и напада (security), пружајући препоруке за спречавање уобичајених софтверских пропуста као што су прекорачење бафера, неконтролисано руковање меморијом и слично. Оба стандарда се широко примењују – MISRA пре свега у аутомобилској и другим безбедносно критичним индустријама, а CERT C у областима где је кључна заштита од сајбер-напада. Важно је нагласити да се MISRA и CERT С не искључују међусобно; напротив, могу се користити комплементарно. Применом MISRA смерница се поставља темељ поузданог и структурно исправног кода, након чега CERT C препоруке додају додатни ниво заштите од злонамерних сценарија, чинећи софтвер и безбедним и сигурним. Поред тога, у пракси се могу срести и други доменски стандарди и препоруке – на пример, ISO 26262 захтева да произвођачи у аутомобилској индустрији користе одговарајуће стандарде кодирања као део процеса обезбеђивања функционалне безбедности, док CERT C допуњује ту причу аспектима сајбер безбедности. У неким организацијама примену налазе и интерни стилски водичи или алтернативни стандарди (попут Barr-C смерница за уграђено програмирање), којима се додатно прецизирају правила кодирања у складу са специфичностима пројекта.

#### 3.3.2. Конзистентност стила и одрживост кода

Осим придржавања формалних стандарда, одржавање конзистентног стила кодирања у целом пројекту има велики утицај на читљивост и одрживост софтвера. Под **стилом програмирања** подразумева се читав скуп правила и навика које код чине једноставним за праћење: конзистентно форматирање (увлачење линија, постављање заграда и размака), смислено именовање променљивих, константи и функција, структуирање кода по логичким целинама, као и писање јасних коментара где год је потребно. Уједначен стил олакшава тимски рад — различити програмери ће брже разумети туђи код ако сви прате исте конвенције. Стилска усклађеност такође поједностављује **code review** поступак (мануелну проверу кода од стране колега) и доприноси смањењу броја грешака у касним фазама развоја.

Стандардизација стила и придржавање договорених смерница данас су саставни део процеса развоја софтвера за микроконтролере. Коришћењем индустријских стандарда као што су MISRA С и CERT С, потпомогнутим алатима за статичку анализу који аутоматски откривају одступања од правила, успоставља се висок ниво квалитета кода. Доследан и добро документован код је не само мање склон грешкама већ је и лакше преносив на нове платформе и одржив током времена. На тај начин, стил програмирања и стандардизација представљају два повезана аспекта квалитета софтвера — први осигурава читљивост и једнообразност, а други уводи проверљива правила која подижу поузданост и безбедност система у целини. Поштовањем ових принципа, развојни тим гради основу за софтвер који ће бити отпоран на грешке, предвидив у понашању и усклађен са строгим захтевима уграђених критичних апликација.

# 3.4. Приступ меморијски мапираним регистрима

#### 3.4.1. Меморијски мапирани улази и излази у микроконтролерима

У типичној *embedded* архитектури, периферни уређаји се контролишу путем меморијски мапираних регистара — посебних хардверских регистара који су изложени у заједничком адресном простору процесора. Део расположивих адреса рачунара резервисан је за ове уређаје, па упис података на одређену меморијску адресу заправо шаље податак периферном уређају, док читање са те адресе доводи до очитавања податка из уређаја. Ово значи да се исте инструкције које СРU користи за приступ обичној меморији (нпр. *load/store* операције) могу користити и за приступ периферијама. Хардверски адресни декодер на системској магистрали препознаје да ли дата адреса припада меморији или уређају и усмерава сигнале и податке ка одговарајућој компоненти. За разлику од тзв. *port-mapped* I/O приступа (карактеристичног за неке раније архитектуре са посебним I/O инструкцијама), меморијски мапиран I/O поједностављује дизајн процесора и омогућава јединствен и ефикасан начин комуникације са уређајима, у складу са RISC филозофијом.

#### 3.4.2. Адресни простор и распоред периферија

Микроконтролери обично имплементирају *Von Neumann* модел меморије са јединственим адресним простором за програмски код, податке и периферије. На пример, ARM Cortex-M архитектура дефинише 4 GB адресног простора подељеног на регионе за Flash (код), SRAM и периферије. Типично је велики блок од 512 MB резервисан за регистре on-chip периферних уређаја – код ARM Cortex-M језгара овај Peripheral регион обухвата адресе отприлике од 0x4000 0000 до 0x5FFF FFFF. У том опсегу смештени су регистри разноврсних модула као што су GPIO, тајмери, UART, A/D конвертори и др; сваки уређај добија сопствени подпојас адреса за своје регистре. Читањем или писањем на било коју адресу у оквиру тог опсега, СРU у ствари приступа одговарајућем регистру периферије. Поред уобичајених периферија, и поједини системски контролни регистри (нпр. регистри за управљање прекидима, тактом или дебагом) такође су мапирани у посебан регион адресног простора – на пример, Private Peripheral Bus регион око адресе 0xE000 0000 код ARM Cortex-M садржи NVIC, SysTick и друге кључне регистре језгра. Овако дефинисана меморијска мапа поједностављује пројектовање boot софтвера и олакшава преносивост програма, јер сва Cortex-M језгра имају сличну организацију адресног простора за основне компоненте система.

#### 3.4.3. Приступ регистрима у програму (С језик)

Са становишта софтвера, рад са меморијски мапираним регистрима своди се на уписивање и читање одређених адреса у меморији. Језик С омогућава веома директан приступ — корисник може декларацијом показивача на дату адресу или коришћењем одговарајућег header-а читати и мењати вредности хардверских регистра као да су променљиве у меморији. Међутим, да би се очувала исправна семантика, неопходно је те променљиве означити као volatile. Кључна реч volatile упозорава компајлер да се вредност дате променљиве може мењати изван тренутног програма (нпр. од стране хардвера или другог thread-а) те да не сме оптимизовати приступе — сваки упис или читање у изворном коду мора резултирати стварним уписом или читањем на датој адреси. У супротном, могло би се десити да компајлер негенерише очекивану инструкцију (нпр. ако "закључи" да се вредност није променила) или да је задржи у регистру процесора, што би нарушило комуникацију са уређајем. Из тог разлога, регистарске константе у header-има микроконтролера увек су декларисане као volatile.

#### 3.4.4. Моделирање хардверских регистра у С

Да би се олакшало коришћење меморијски мапираних регистра, у пракси се примењује техника мапирања регистара на С структуре. Идеја је да се дефинише typedef struct чија поља тачно одговарају регистрима једног периферног модула редом којим су они распоређени у меморијском простору. Затим се креира показивач (или макро) на ту структуру на базној адреси периферије. На тај начин, сваки регистар се може именовано адресирати преко поља структуре уместо преко "магичних" хексадецималних константи. Ознака volatile се може применити на саму структуру или на показивач, чиме се гарантује да ће сваки приступ пољима структуре заиста приступити физичком регистру. Практично сваки савремени произвођач микроконтролера уз своје уређаје испоручује и одговарајуће заглавље са већ унапред дефинисаним структурама и базним адресама периферија. ARM је стандардизовао овај приступ кроз CMSIS (Cortex Microcontroller Software Interface Standard), па се у CMSIS header-има налазе описне структуре и макрои за све регистре циљаног система. На пример, у наставку је приказана поједностављена дефиниција једног GPIO модула и коришћење његових регистара:

Горњи код илуструје принцип меморијски мапираног приступа периферији. Структура GPIO ТуреDef декларативно описује низ од три 32-битна регистра – замислимо да су то улазни, излазни и регистар правца GPIO порта. Макро GPIO дефинише показивач на ову структуру на меморијској адреси 0x50000000, за коју претпостављамо да је базна адреса одговарајућег GPIO контролера у датом микроконтролеру. Када у програму извршимо наредбу GPIO->OUT = 0x1;, компајлер ће генерисати машинску инструкцију за упис вредности 1 на меморијску адресу која одговара регистру *OUT* тог модула (нпр. инструкцију STR на ARM архитектури). Овим уписом се хардверски излаз на пину 0 поставља на високи ниво (под условом да је тај пин претходно конфигурисан као излаз, као у примеру где се GPIO->DIR подешава). Читљивост је знатно побољшана – уместо неразумљивог израза \*(volatile uint32 t \*)(0x50000004) = 0x1; који директно адресира меморију, програмер користи симболичко име GPIO->OUT, што јасно означава шта се догађа. Савремени преводиоци ће овакву употребу структура оптимизовати једнако ефикасно као и коришћење директних показивача или макроа; резултујући машински код је идентичан, па нема казне у погледу перформанси. Дакле, главна разлика је у побољшаној прегледности и типској безбедности кода, без жртвовања ефикасности.

#### 3.4.5. Предности и значај апстракције регистра

Стандарди попут CMSIS-а и званични header-и произвођача обезбеђују да програмери не морају ручно да дефинишу сваки регистар и адресу — већ су им на располагању унапред проверене дефиниције структуре и базних адреса. Ово смањује могућност грешке и унапређује преносивост софтвера између различитих платформи. Код написан уз коришћење симболичких регистара (нпр. RCC->AHB1ENR или GPIOD->ODR у случају STM32 микроконтролера) много је разумљивији него код са "магичним" бројевима адреса, што доприноси бољој одрживости. У академском и индустријском контексту, овакав ниво апстракције се препоручује као део добрих пракси пројектовања: повећава се кохезија и јасно раздвајање надлежности софтверских модула, чинећи систем лакшим за верификацију и одржавање. На крају, приступ меморијски мапираним регистрима представља основни механизам којим bare-metal фирмвер остварује интеракцију са физичким светом — кроз промишљено коришћење овог механизма, постиже се детерминистичко, брзо и предвидиво извршавање управљачког кода, што је од пресудне важности за реалновременске примене.

# 4. Фазе компилације (превођења) у GCC

Превођење С програма у машински код одвија се кроз више дискретних фаза. GCC компајлер (GNU Compiler Collection) интерно дели процес на четири корака: препроцесирање, компилацију (у ужем смислу), асемблирање и линковање, тим редоследом. Свака фаза има своју улогу у претварању изворног . С кода у извршну бинарну датотеку. Следи преглед ових фаза у табели 1.

Табела 1. Фазе превођења С програма уз GCC

| Фаза                 | Алат (GCC позив) | Улаз                  | Излаз                    |
|----------------------|------------------|-----------------------|--------------------------|
| Препроцесирање       | gcc -E           | *.c, *.h (изворник)   | Препроцесирани код (*.i) |
| Компилација (C->ASM) | gcc -S           | *.і (из претходног)   | Асемблерски код (*.s)    |
| Асемблирање          | gcc -сили as     | *.s (асемблерски код) | Објектни фајл (*.0)      |
| Линковање            | gcc или ld       | *.о (+ библиотеке)    | Извршна датотека (ELF)   |

# 4.1. Препроцесирање

Препроцесирање је прва фаза у којој се извршава С препроцесор. Он обрађује директиве које почињу знаком # — на пример, убацује садржај хедер датотека на место #include директива, проширује макрое дефинисане са #define, и условно уклања/укључује делове кода на основу #ifdef услова. Резултат ове фазе је препроцесирани исходни код, типично са екстензијом .i (или .ii за C++) који више не садржи препроцесорске директиве, већ само "чист" С код.

## 4.2. Компилација

У фази компилације (у ужем смислу) GCC преводи препроцесирани С код у асемблерски код за циљну архитектуру. То значи да се синтакса и конструкције С језика преводе у низ асемблерских инструкција (нпр. ARM Cortex-M7 инструкције) које остварују еквивалентну функционалност. Излаз из ове фазе је .s датотека (асемблерски код у текстуалном облику). Ова фаза укључује и различите оптимизације које компајлер примењује (према подешеним опцијама, нпр. -O2) како би генерисани код био што ефикаснији.

# 4.3. Асемблирање

Генерисани .s асемблерски код затим пролази кроз асемблер (саставни део GCC алата, нпр. аrm-none-eabi-as), који га претвара у *релокативни објектни фајл* – машински код са нерешеним релокацијама и симболима. Ова датотека обично има екстензију .o и у формату је објектне датотеке (најчешће ELF формат, о чему ће бити речи касније). Објектни фајл садржи машинске инструкције за дати модул, али још увек није самостално извршна целина, јер адресе функција и података који се налазе у другим модулима нису још познате (остају као симболи које треба повезати).

# 4.4. Линковање

Последња фаза је позив линкера (нпр. GNU ld) који узима један или више објектних фајлова (.о), као и евентуално предефинисане библиотеке (нпр. libc, или драјверске библиотеке), и **повезује** их у јединствену извршну датотеку. Линкер разрешава све међусобне референце — нпр. када функција у main.о зове функцију која је имплементирана у uart.о, линкер ће уписати исправну адресу те функције у машински код позива. Такође, линкер припаја и стандардни стартуп код (нпр. **crt0** за C) ако је део toolchain-а, мада у embedded окружењу стартуп и векторска табела обично долазе као засебан модул пројекта. Резултат линковања је извршна бинарна датотека у *ELF формату* (или сличном), са свим спојеним секцијама на одговарајућим меморијским адресама. Ова датотека је сада самосталан програм који се може учитати у меморију микроконтролера

## 4.5. Интеграција фаза компилације у оквиру GCC алатке

Напоменимо да се у пракси већина ових корака обавља "у пролазу" помоћу исте gcc наредбе, јер GCC аутоматски позива препроцесор, па компајлер, асемблер и линкер. На пример, позив

arm-none-eabi-gcc -O2 -mcpu=cortex-m7 -o program.elf main.c uart.c startup.s -T linker.ld

ће обавити све кораке и произвести коначни program.elf. Ипак, корисно је разумети ове међукоре, јер алати омогућавају да се сваки корак изведе одвојено (нпр. опција -savetemps чува привремене .i и .s датотеке). GCC документација наглашава постојање наведене четири фазе и одговарајуће суфиксе/екстензије фајлова. Током компилације могу настати и помоћни фајлови као што је *пистинг* (са мешовитим С кодом и асемблером, ако је затражено), али они нису нужни у даљем процесу.

Важно је истаћи да линкер за успешно повезивање за *embedded* мету мора знати распоред меморије циљног микроконтролера — ту ступа на снагу *линкерска скрипта* која описује меморијске регије (Flash, RAM) и како распоредити секције програма у њих. Линкерска скрипта је критична за добијање исправне бинарне слике програма и њена структура биће детаљно анализирана у посебном одељку.

# 5. Формати резултујућих датотека

Увод.

## 5.1. ELF формат

По завршеном линковању, добија се извршна датотека, најчешће у **ELF формату** (Executable and Linkable Format). ELF је стандардни бинарни формат који се користи на Unix/Linux системима за извршне датотеке, објектне модуле, па чак и библиотеке. Он је прихваћен и код cross-компајлера за микроконтролере јер је флексибилан и независан од архитектуре – подржава различите процесоре, ендијаност и величине адресног простора. За потребе embedded програмирања, ELF садржи све потребне информације о програму: машински код сегментиран у секције (.text, .data, .bss, итд.), али и симболичке табеле, таблице релокација, програмска заглавља са описом сегмената за извршавање, и опционе дебаг информације. ELF формат подржава двоструку анализу: према табели секција (section header table), која описује структуру изворног кода и симболе, или према табели сегмената (program header table), која описује начин учитавања у меморију при извршавању. У контексту микроконтролера, важнији је распоред секција, јер сегменти одговарају меморијским регијама у које ће секције бити смештене (Flash, RAM).

Иако ELF датотека садржи извршни код, она се обично не програмира директно у микроконтролер. Разлог је што ELF носи и метаподатке (нпр. симболе, одређене секције које нису потребне за сам рад програма) и није у формату који типични програмабилни хардвер очекује. Зато се из ELF-а изводе *чисти бинарни формати* погодни за флешовање. Најчешће се срећу три таква формата у embedded свету: Intel HEX, RAW BIN (сиров бинарни фајл), и Motorola S-Record (S19) формат.

## 5.2. Intel HEX формат

Intel HEX формат је текстуални формат у ASCII нотацији који представља садржај меморије у хексадецималном облику. Датотека се састоји од више линија, где свака линија представља један рекорд са одређеним бројем бајтова, њиховом адресом у меморији и контролном сумом. Конкретно, свака линија почиње двотачком, затим следи бајт бројача (колико бајтова података та линија носи), па 16-битна почетна адреса, бајт типа записа (нпр. 00 за податке, 01 за крај датотеке, 04 за проширену адресу код већих адресних простора итд.), затим сами подаци (парови хекс цифара), и на крају једна контролна сума за проверу тачности. Овај формат је веома погодан јер је читљив и садржи адресе – нпр. ако програм није континуиран у меморији, НЕХ фајл може имати "рупе" у адресама између линија. Програматори (alati за флешовање) читају НЕХ датотеку линију по линију и уписују бајтове на наведене адресе у флеш меморију микроконтролера. Intel HEX је историјски настао 1970-их за потребе учитавања програма са папирне траке у Intel MCS systems, али је и данас широко коришћен због једноставности и поузданости провере (свака линија носи своју контролну суму). Генерисање НЕХ фајла се обично ради алатом овјсору, о чему ће бити речи касније.

## 5.3. RAW бинарни формат (.bin)

RAW бинарни формат (.bin) је најједноставнији могући формат — низ бајтова идентичан бајтовима који треба да се упишу у меморију, без икакве додатне структуре или информација. Сирови бинарни фајл представља *меморијски дамп* програма, обично тачно оних секција које се налазе у непрекидном опсегу адреса. При конверзији ELF-а у .bin, одбацују се сви симболи, заглавља и вишак информација, и добија се само секвенца бајтова која одговара садржају флеша (и евентуално других меморија ако се спајају у један фајл). GNU objcopy алат омогућава ову конверзију: на пример команда

#### arm-none-eabi-objcopy -O binary program.elf program.bin

узима ELF и ствара .bin фајл. Према документацији, када се објсору користи за генерисање гаw binary датотеке, он ефективно производи меморијски дамп укупног садржаја ELF-а — од најнижег до највишег адресног бајта садржаног у ELF-у — одбацујући све симболе и релокације. Битно је напоменути да .bin не носи информацију о томе на коју адресу ти бајтови треба да се упишу; претпоставља се подразумевани почетак (нпр. код већине микроконтролера почетак флеша је или 0х00000000 или нека позната базна адреса). Због тога, .bin формат се углавном користи када се читава слика програма ставља на почетак флеш меморије. Ако је потребно флешовати програм који не почиње од 0 или ако програм обухвата више одвојених меморијских области, Intel HEX је погоднији, јер носи адресе.

У пракси, многи произвођачи алата и IDE-ови (нпр. KEIL uVision, IAR EWARM, GCC toolchain) омогућавају генерисање HEX или BIN датотека из ELF-а. Неки *debugger*-и и *програматори* могу чак директно учитати ELF (користећи информације из ELF заглавља о сегментима за учитавање). Ипак, имајући у виду да HEX и BIN представљају стандард у размену фирмвер слика (нпр. HEX за надоградњу софтвера у сервису, или BIN за брзо учитавање преко bootloader-а), важно је разумети њихову структуру и разлике.

## 5.4. Motorola S-Record формат (S19)

Поред Intel HEX-а, чест је и **Motorola S-Record (S19)** формат – сличан ASCII хекс запису линија. Alati **objcopy** са опцијом **-O srec** може генерисати S-Record фајл. Разлика је углавном у синтакси линија (S-Record линије почињу са 'S' и имају мало другачију организацију адресних поља). Пошто је у питању алтернативни формат, нећемо детаљно разматрати његову структуру, али вреди споменути да алат **srec\_cat** (део SRecord пакета) може манипулисати и HEX и S19 фајловима.

# 6. Употреба GNU алата: objcopy, readelf, nm, size, objdump

При раду на озбиљним embedded пројектима, корисно је познавати алате за анализу и конверзију објектних и извршних датотека. GCC toolchain долази са низом **GNU binutils** алата који омогућавају увид у ELF садржај, конверзију формата, дисасемблирање, мерење величине и друго. У наставку су описани најважнији алати и њихова примена у контексту програмигања микроконтролера.

## 6.1. objcopy

Ово је алат за копирање и конверзију формата објектних датотека. Користи се за генерисање HEX или BIN фајла из ELF-а. Примери:

objcopy -O ihex program.elf program.hex (створи Intel HEX из ELF-a),

objcopy -O binary program.elf program.bin (створи raw бинарни дамп).

Овај алат може издвајати и поједине секције, уклањати debug информације опцијом -S, итд. Базиран је на BFD библиотеци, тако да подржава бројне формате и аутоматски препознаје улазни и излазни формат. Треба бити опрезан: при конверзији у .bin, ако ELF садржи "празнину" (нпр. секције које нису континуалне у меморији), оbјсору ће ту празнину испунити нулама у .bin фајлу (јер прави континуални дамп од најнижег до највишег адресног бајта). Стога .bin може бити већи него износ корисног кода ако постоје велике неиницијализоване секције на високим адресама.

#### 6.2. readelf

Алат **readelf** служи за читање ELF датотека – даје детаљан увид у ELF заглавља, секције, сегменте, симболе, итд. На пример, *readelf -h program.elf* приказује опште заглавље (тип, машина, ендијаност, улазну тачку...), *readelf -S program.elf* листа секције (са именима, величинама, стартним адресама у фајлу и у меморији), а *readelf -s program.elf* листа симбол таблицу (са именима функција/променљивих и њиховим адресама или офсетовима). У анализи линкер скрипте и меморијског распореда, *readelf -S* је посебно користан да видимо где су .text, .data, .bss и друге секције смештене и колике су. **Пример:** излаз *readelf -S* може показати да је .text секција величине, 0x1000 бајтова на адреси 0x10000000 (у флешу), .data величине 0x100 бајтова на адреси 0x08040000 (RAM), са LMA (Load Memory Address) у флешу, итд. Ово нам јасно говори распоред по меморијским

#### 6.3. nm

Овај алат служи за листање симбола (из објектних или извршних фајлова). пт program.elf ће исцртати листу свих симбола (функција, глобалних променљивих) које постоје у програму, са њиховим адресама и ознаком типа (Т=текст/функција, D=иницијализовани податак, B=неиницијализовани податак bss, и сл.). Ово је корисно кад желимо да знамо на којој адреси се налази одређена функција или променљива након линковања. На пример, можемо проверити да ли је глобална променљива мапирана у RAM (биће означена са В или D и имаће адресу у опсегу RAM меморије), или да ли је функција у флешу (ознака Т са адресом у опсегу флеша). nm је посебно драгоцен за грубу проверу исправности линкер скрипте – нпр. да ли симбол estack (врх стека) има очекивану симболи преклопљени вредност, ла ЛИ cy неки

#### 6.4. size

Овај алат приказује резиме величина секција у извршној датотеци. Обично се позива као size program.elf и избацује три колоне: величину .text (код + константни подаци у флешу), .data (иницијални подаци који ће бити учитани у RAM) и .bss (неиницијализовани подаци који ће заузети RAM), као и укупан збир. Ово је врло прегледно да се види колики је "отисак" програма у флешу и RAM-у. На пример, output може бити:

| text | data | bss | dec  | hex  | filename    |
|------|------|-----|------|------|-------------|
| 4528 | 128  | 256 | 4912 | 1330 | program.elf |

што значи да код заузима  $\sim$ 4528 бајтова флеша, статички иницијализовани подаци 128 бајтова RAM-а (плус још толико у флешу за њихове почетне вредности), а .bss (нпр. глобалне променљиве иницијализоване на 0) заузимају 256 бајтова RAM-а. Ови бројеви су битни у планирању да ли програм стаје у меморију микроконтролера. (Напомена: size уз опцију -A или --format=SysV даје детаљнији приказ по именованим секцијама.)

## 6.5. objdump

objdump је многофункционални алат за испис садржаја објектних датотека. Може да прикаже хексадецимални dump секција (objdump -s program.elf), али најкориснија функционалност је дисасемблирање машинског кода у читљив асемблер.

Наредба *objdump -d -M reg-names-std program.elf* произвешће асемблерски листинг свих секција које имају код (нпр. .text, .init) са људски читљивим именима регистара. Ово је изузетно корисно за дебаговање ниског нивоа – можемо видети тачно које је инструкције компајлер генерисао из нашег С кода, што помаже у оптимизацији или трагању за баговима на ниском нивоу.

objdump -t program.elf исписује таблицу симбола (слично nm).

objdump -x program.elf исписује пуна ELF заглавља, секције, сегменте, симболе (комбинација информација, мање читљива од специјализованих алата попут readelf или nm).

Углавном, *objdump* је згодан за брзи увид у садржај бинарног кода – било у хекс или у асемблерском облику.

#### 6.6. Резиме

Набројани алати покривају најважније аспекте: конверзију формата (objcopy, srec\_cat), анализу садржаја (readelf, nm, objdump) и величину (size). У типичном развојном циклусу, након добијања program.elf, програмер може покренути size да провери заузеће меморије, objdump -d ако сумња у неку оптимизацију компајлера, или nm да пронађе адресу битне променљиве за дебаговање. При припреми НЕХ-а за програмирање, користи се objcopy или srec\_cat. На тај начин, ови алати представљају продужетак функционалности самог компајлера и линкера, дајући увид "испод хаубе" готовог програма.

# 7. Линкерска скрипта: MEMORY и SECTIONS дефиниције

Линкерска скрипта или линкер директива (.ld датотека) је суштински део embedded пројекта — она диктира како ће линкер распоредити преведене секције програма у меморијски адресни простор микроконтролера. Пошто микроконтролер има одређене величине и адресе флеш и RAM меморије, потребно је линкер обавестити где може смештати код, а где податке. GNU ld линкер има свој интерни језик за скрипте који омогућава опис меморијских региона и расподелу секција.

#### 7.1. MEMORY дефиниција

У скрипти се најпре дефинишу именовани *меморијски региони*. Пример за неки Cortex-M7 микроконтролер може бити:

```
MEMORY

{

FLASH (rx) : ORIGIN = 0x10000000, LENGTH = 4096K

WORK_FLASH (rx) : ORIGIN = 0x14000000, LENGTH = 256K

SRAM (rwx) : ORIGIN = 0x28000000, LENGTH = 768K

}
```

Овим смо линкеру дали до знања да постоје три региона: FLASH који почиње на адреси 0x1000\_0000 величине 4 МВ, додатни WORK\_FLASH од 256 КВ, и SRAM од 768 КВ почев од 0x2800\_0000. Сваки регион има атрибуте: r (читљив), w (уписив), x (извршан). Флеш је обично означен са rx (читање + извршавање, није уписив у току рада програма), RAM са rwx (читање/писање, извршавање ако се код копира у RAM). Ове МЕМОRУ декларације омогућавају линкеру да разуме расположиви адресни простор и да пријави грешку ако одређена секција премашује додељени меморијски блок. На пример, ако код постане превелик (прелази 4МВ флеша), линкер ће јавити да FLASH регион нема довољно простора. МЕМОRУ секција дакле описује распон и намену меморијских блокова циљног система.

#### 7.2. SECTIONS расподела

Након дефинисања меморије, линкерска скрипта садржи SECTIONS блок, који је срце скрипте – ту се наводи како се свакој излазној секцији ELF-а додељује меморијски регион и адреса. Пример делимичне SECTIONS скрипте за ARM:

```
SECTIONS {
    .text:
   {
                          /* почетак .text секције */
      _stext = .;
      KEEP(*(.isr\_vector)) /* векторска табела, ако је дефинисана у секцији
.isr_vector */
                          /* све .text секције из свих објеката */
      *(.text*)
      *(.rodata*)
                          /* све .rodata (константе) секције */
      _etext = .;
                       /* крај .text/.rodata секције */
                          /* овај блок иде у FLASH меморију */
   } > FLASH
    .data : AT (ADDR(FLASH) + SIZEOF(.text)) /* AT(...) каже да се почетни
садржај .data налази након .text y FLASH */
                          /* почетак .data y RAM */
       _sdata = .;
                          /* све иницијализоване статичке променљиве */
      *(.data*)
      _edata = .;
   } > SRAM
                           /* овај блок се мапира у SRAM на извршавање */
    .bss (NOLOAD) :
      _{sbss} = .;
                          /* почетак .bss */
      *(.bss*)
      *(COMMON)
      _ebss = .;
                           /* .bss je y SRAM, NOLOAD јер нема почетних
   } > SRAM
података у флешу */
   /* ... остале секције (стек, heap, секције за C++ EH, итд.) ... */
```

Овај пример илуструје кључне концепте. Прво, .text (са .rodata) је смештен у FLASH регион. У њему се чак принудно чува (КЕЕР) векторска табела ако је дефинисана у посебној секцији .isr\_vector (многи startup кодови стављају векторску таблицу у засебну секцију на почетак флеша). Затим .text обухвата сав програмски код и неизменљиве податке. Симболи као \_stext и \_etext су обележивачи које можемо касније користити (нпр. \_etext указује на крај садржаја који се налази у флешу, што је уједно почетак података за копирање у RAM).

Део за .data је интересантан јер демонстрира одвајање LOAD adpece и RUN adpece: секција .data ће током извршавања бити у SRAM (због > SRAM), али је наведено AT (ADDR(FLASH) + SIZEOF(.text)). Ово каже линкеру да резервише простор у FLASH-у за иницијалне вредности .data секције, одмах након .text блока. Практично, то значи да ће

у ELF-у .data имати два различита почетка: VMA (Virtual Memory Address) у RAM (нпр.  $0x2800\_0000$  ако ту почиње SRAM) и LMA (Load Memory Address) у флешу (нпр.  $0x1000\_1234$  ако .text заузима до  $0x1000\_1233$ ). Линкер ће поред ELF секције уписати и тај садржај (иницијалне бајтове) на LMA адресу. Потом, стартуп код микроконтролера копира те бајтове из флеша на одговарајућу VMA адресу у SRAM током покретања програма. На овај начин, глобалне променљиве које имају иницијалне вредности (нпр. int count = 5;) налазе се у .data секцији: након ресета, у RAM-у на адреси \_sdata треба да се нађе вредност 5, коју је програм негде морао сачувати — управо у флешу, у склопу .data LMA. Директива AT() у линкер скрипти се користи да раздвоји те адресе и веома је битна за исправно покретање програма.

Секција .bss у примеру има атрибут NOLOAD – то значи да за њу не постоји садржај у флешу; она представља блок RAM меморије коју треба испунити нулама. Линкерски симболи \_sbss и \_ebss означавају почетак и крај .bss. Важно је и да су сви СОММОN симболи (неиницијализоване глобалне променљиве које нису експлицитно у .bss) такође стављени у .bss (директива \*(СОММОN)). Стартуп код ће једноставно обрисати (испунити нулама) овај RAM опсег при покретању.

Поред ових, линкер скрипта често резервише простор за стек (нпр. дефинише симбол \_estack на крај SRAM-а), за heap (ако се користи), и укључује специјалне секције за C++ ако су присутне (нпр. .init\_array за конструкторе објеката). У случају ARM Cortex-M, потребне су и секције .ARM.exidx и .ARM.extab за информације о изузецима (корисно ако се користе одређене библиотеке и функције или C++ бацање изузетака – throw exception). Они се обично смештају на крај .text сегмента. Укључујући све те ставке, линкерска скрипта обезбеђује да у излазном ELF-у стоје исправно дефинисане адресе и границе секција.

За пример TRAVEO T2G микроконтролера, линкерска скрипта би била прилагођена тачним величинама меморија тог чипа. На основу **меморијске архитектуре**, могли бисмо дефинисати MEMORY са

FLASH (rx) : ORIGIN = 0x100000000, LENGTH = 4160K

WORKFLASH (rx): ORIGIN = 0x14000000, LENGTH = 256K

SRAM (rwx) : ORIGIN = 0x28000000, LENGTH = 768K

, и можда посебно ITCM и DTCM меморије ако би се користиле (16KB + 16KB по језгру). За једноставност, претпоставимо да сав код иде у главни FLASH, а сав RAM садржај у главни SRAM. Линкерска скрипта би онда одразила тај модел: .text, .rodata, евентуално векторска табела и startup код у FLASH; .data LMA у FLASH али VMA у SRAM; .bss у

SRAM.

Закључак о линкер скрипти: Она повезује свет С кода са физичком меморијом хардвера. МЕМОRY секција описује где може шта да иде, а SECTIONS секција како да се садржај распореди. За типичан Cortex-M пројекат, већина програмера користи унапред припремљену скрипту (било од произвођача или генерисану алатом), али разумевање исте је кључно при решавању проблема као што су: преливање меморије, смештање специфичних функција у одређени регион (нпр. у посебан сегмент који се може ажурирати независно), прављење боотлоадер/апликација поделе, итд.

8. Компилација и меморијски распоред за Infineon TRAVEO T2G

# 9. Закључак